home *** CD-ROM | disk | FTP | other *** search
/ PC World Interactive 7 / PC World Interactive 7.iso / share / multimed / myflix_win32 / myflix_win32.exe / data1.cab / Libraries / tcl8.0 / Safe.tcl < prev    next >
Text File  |  1998-03-10  |  21KB  |  711 lines

  1. # safe.tcl --
  2. #
  3. # This file provide a safe loading/sourcing mechanism for safe interpreters.
  4. # It implements a virtual path mecanism to hide the real pathnames from the
  5. # slave. It runs in a master interpreter and sets up data structure and
  6. # aliases that will be invoked when used from a slave interpreter.
  7. # See the safe.n man page for details.
  8. #
  9. # Copyright (c) 1996-1997 Sun Microsystems, Inc.
  10. #
  11. # See the file "license.terms" for information on usage and redistribution
  12. # of this file, and for a DISCLAIMER OF ALL WARRANTIES.
  13. #
  14. # SCCS: @(#) safe.tcl 1.21 97/08/13 15:37:22
  15.  
  16. #
  17. # The implementation is based on namespaces. These naming conventions
  18. # are followed:
  19. # Private procs starts with uppercase.
  20. # Public  procs are exported and starts with lowercase
  21. #
  22.  
  23. # Needed utilities package
  24. package require opt 0.1;
  25.  
  26. # Create the safe namespace
  27. namespace eval ::safe {
  28.  
  29.     # Exported API:
  30.     namespace export interp \
  31.         interpAddToAccessPath interpFindInAccessPath \
  32.         setLogCmd ;
  33.  
  34. # Proto/dummy declarations for auto_mkIndex
  35. proc ::safe::interpCreate {} {}
  36. proc ::safe::interpInit {} {}
  37. proc ::safe::interpConfigure {} {}
  38. proc ::safe::interpDelete {} {}
  39.  
  40.  
  41.     # Interface/entry point function and front end for "Create"
  42.     ::tcl::OptProc interpCreate {
  43.     {?slave? -name {} "name of the slave (optional)"}
  44.     {-accessPath -list {} "access path for the slave"}
  45.     {-noStatics "prevent loading of statically linked pkgs"}
  46.     {-nestedLoadOk "allow nested loading"}
  47.     {-deleteHook -script {} "delete hook"}
  48.     } {
  49.     InterpCreate $slave $accessPath \
  50.         [expr {!$noStatics}] $nestedLoadOk $deleteHook;
  51.     }
  52.  
  53.     # Interface/entry point function and front end for "Init"
  54.     ::tcl::OptProc interpInit {
  55.     {slave -name {} "name of the slave"}
  56.     {-accessPath -list {} "access path for the slave"}
  57.     {-noStatics "prevent loading of statically linked pkgs"}
  58.     {-nestedLoadOk "allow nested loading"}
  59.     {-deleteHook -script {} "delete hook"}
  60.     } {
  61.     InterpInit $slave $accessPath \
  62.         [expr {!$noStatics}] $nestedLoadOk $deleteHook;
  63.     }
  64.  
  65.     # Interface/entry point function and front end for "Configure"
  66.     ::tcl::OptProc interpConfigure {
  67.     {slave -name {} "name of the slave"}
  68.     {-accessPath -list {} "access path for the slave"}
  69.     {-noStatics "prevent loading of statically linked pkgs"}
  70.     {-nestedLoadOk "allow nested loading"}
  71.     {-deleteHook -script {} "delete hook"}
  72.     } {
  73.     # Check that at least one flag was given:
  74.     if {[string match "*-*" $Args]} {
  75.         # reconfigure everything (because otherwise you can't
  76.         # change -noStatics for instance)
  77.         InterpConfigure $slave $accessPath \
  78.             [expr {!$noStatics}] $nestedLoadOk $deleteHook;
  79.         # auto_reset the slave (to completly synch the new access_path)
  80.         if {[catch {::interp eval $slave {auto_reset}} msg]} {
  81.         Log $slave "auto_reset failed: $msg";
  82.         }
  83.     } else {
  84.         # none was given, lets return current values instead
  85.         set res {}
  86.         lappend res [list -accessPath [Set [PathListName $slave]]]
  87.         if {![Set [StaticsOkName $slave]]} {
  88.         lappend res "-noStatics"
  89.         }
  90.         if {[Set [NestedOkName $slave]]} {
  91.         lappend res "-nestedLoadOk"
  92.         }
  93.         lappend res [list -deleteHook [Set [DeleteHookName $slave]]]
  94.         join $res
  95.     }
  96.     }
  97.  
  98.  
  99.     #
  100.     # safe::InterpCreate : doing the real job
  101.     #
  102.     # This procedure creates a safe slave and initializes it with the
  103.     # safe base aliases.
  104.     # NB: slave name must be simple alphanumeric string, no spaces,
  105.     # no (), no {},...  {because the state array is stored as part of the name}
  106.     #
  107.     # Returns the slave name.
  108.     #
  109.     # Optional Arguments : 
  110.     # + slave name : if empty, generated name will be used
  111.     # + access_path: path list controlling where load/source can occur,
  112.     #                if empty: the master auto_path will be used.
  113.     # + staticsok  : flag, if 0 :no static package can be loaded (load {} Xxx)
  114.     #                      if 1 :static packages are ok.
  115.     # + nestedok: flag, if 0 :no loading to sub-sub interps (load xx xx sub)
  116.     #                      if 1 : multiple levels are ok.
  117.     
  118.     # use the full name and no indent so auto_mkIndex can find us
  119.     proc ::safe::InterpCreate {
  120.     slave 
  121.     access_path
  122.     staticsok
  123.     nestedok
  124.     deletehook
  125.     } {
  126.     # Create the slave.
  127.     if {[string compare "" $slave]} {
  128.         ::interp create -safe $slave;
  129.     } else {
  130.         # empty argument: generate slave name
  131.         set slave [::interp create -safe];
  132.     }
  133.     Log $slave "Created" NOTICE;
  134.  
  135.     # Initialize it. (returns slave name)
  136.     InterpInit $slave $access_path $staticsok $nestedok $deletehook;
  137.     }
  138.  
  139.  
  140.     #
  141.     # InterpConfigure (was setAccessPath) :
  142.     #    Sets up slave virtual auto_path and corresponding structure
  143.     #    within the master. Also sets the tcl_library in the slave
  144.     #    to be the first directory in the path.
  145.     #    Nb: If you change the path after the slave has been initialized
  146.     #    you probably need to call "auto_reset" in the slave in order that it
  147.     #    gets the right auto_index() array values.
  148.  
  149.     proc ::safe::InterpConfigure {slave access_path staticsok\
  150.         nestedok deletehook} {
  151.  
  152.     # determine and store the access path if empty
  153.     if {[string match "" $access_path]} {
  154.         set access_path [uplevel #0 set auto_path];
  155.         # Make sure that tcl_library is in auto_path
  156.         # and at the first position (needed by setAccessPath)
  157.         set where [lsearch -exact $access_path [info library]];
  158.         if {$where == -1} {
  159.         # not found, add it.
  160.         set access_path [concat [list [info library]] $access_path];
  161.         Log $slave "tcl_library was not in auto_path,\
  162.             added it to slave's access_path" NOTICE;
  163.         } elseif {$where != 0} {
  164.         # not first, move it first
  165.         set access_path [concat [list [info library]]\
  166.             [lreplace $access_path $where $where]];
  167.         Log $slave "tcl_libray was not in first in auto_path,\
  168.             moved it to front of slave's access_path" NOTICE;
  169.         
  170.         }
  171.  
  172.         # Add 1st level sub dirs (will searched by auto loading from tcl
  173.         # code in the slave using glob and thus fail, so we add them
  174.         # here so by default it works the same).
  175.         set access_path [AddSubDirs $access_path];
  176.     }
  177.  
  178.     Log $slave "Setting accessPath=($access_path) staticsok=$staticsok\
  179.         nestedok=$nestedok deletehook=($deletehook)" NOTICE;
  180.  
  181.     # clear old autopath if it existed
  182.     set nname [PathNumberName $slave];
  183.     if {[Exists $nname]} {
  184.         set n [Set $nname];
  185.         for {set i 0} {$i<$n} {incr i} {
  186.         Unset [PathToken $i $slave];
  187.         }
  188.     }
  189.  
  190.     # build new one
  191.     set slave_auto_path {}
  192.     set i 0;
  193.     foreach dir $access_path {
  194.         Set [PathToken $i $slave] $dir;
  195.         lappend slave_auto_path "\$[PathToken $i]";
  196.         incr i;
  197.     }
  198.     Set $nname $i;
  199.     Set [PathListName $slave] $access_path;
  200.     Set [VirtualPathListName $slave] $slave_auto_path;
  201.  
  202.     Set [StaticsOkName $slave] $staticsok
  203.     Set [NestedOkName $slave] $nestedok
  204.     Set [DeleteHookName $slave] $deletehook
  205.  
  206.     SyncAccessPath $slave;
  207.     }
  208.  
  209.     #
  210.     #
  211.     # FindInAccessPath:
  212.     #    Search for a real directory and returns its virtual Id
  213.     #    (including the "$")
  214. proc ::safe::interpFindInAccessPath {slave path} {
  215.     set access_path [GetAccessPath $slave];
  216.     set where [lsearch -exact $access_path $path];
  217.     if {$where == -1} {
  218.         return -code error "$path not found in access path $access_path";
  219.     }
  220.     return "\$[PathToken $where]";
  221.     }
  222.  
  223.     #
  224.     # addToAccessPath:
  225.     #    add (if needed) a real directory to access path
  226.     #    and return its virtual token (including the "$").
  227. proc ::safe::interpAddToAccessPath {slave path} {
  228.     # first check if the directory is already in there
  229.     if {![catch {interpFindInAccessPath $slave $path} res]} {
  230.         return $res;
  231.     }
  232.     # new one, add it:
  233.     set nname [PathNumberName $slave];
  234.     set n [Set $nname];
  235.     Set [PathToken $n $slave] $path;
  236.  
  237.     set token "\$[PathToken $n]";
  238.  
  239.     Lappend [VirtualPathListName $slave] $token;
  240.     Lappend [PathListName $slave] $path;
  241.     Set $nname [expr $n+1];
  242.  
  243.     SyncAccessPath $slave;
  244.  
  245.     return $token;
  246.     }
  247.  
  248.     # This procedure applies the initializations to an already existing
  249.     # interpreter. It is useful when you want to install the safe base
  250.     # aliases into a preexisting safe interpreter.
  251.     proc ::safe::InterpInit {
  252.     slave 
  253.     access_path
  254.     staticsok
  255.     nestedok
  256.     deletehook
  257.     } {
  258.  
  259.     # Configure will generate an access_path when access_path is
  260.     # empty.
  261.     InterpConfigure $slave $access_path $staticsok $nestedok $deletehook;
  262.  
  263.     # These aliases let the slave load files to define new commands
  264.  
  265.     # NB we need to add [namespace current], aliases are always
  266.     # absolute paths.
  267.     ::interp alias $slave source {} [namespace current]::AliasSource $slave
  268.     ::interp alias $slave load {} [namespace current]::AliasLoad $slave
  269.  
  270.     # This alias lets the slave have access to a subset of the 'file'
  271.     # command functionality.
  272.  
  273.     AliasSubset $slave file file dir.* join root.* ext.* tail \
  274.         path.* split
  275.  
  276.     # This alias interposes on the 'exit' command and cleanly terminates
  277.     # the slave.
  278.  
  279.     ::interp alias $slave exit {} [namespace current]::interpDelete $slave
  280.  
  281.     # The allowed slave variables already have been set
  282.     # by Tcl_MakeSafe(3)
  283.  
  284.  
  285.     # Source init.tcl into the slave, to get auto_load and other
  286.     # procedures defined:
  287.  
  288.     # We don't try to use the -rsrc on the mac because it would get
  289.     # confusing if you would want to customize init.tcl
  290.     # for a given set of safe slaves, on all the platforms
  291.     # you just need to give a specific access_path and
  292.     # the mac should be no exception. As there is no
  293.     # obvious full "safe ressources" design nor implementation
  294.     # for the mac, safe interps there will just don't
  295.     # have that ability. (A specific app can still reenable
  296.     # that using custom aliases if they want to).
  297.     # It would also make the security analysis and the Safe Tcl security
  298.     # model platform dependant and thus more error prone.
  299.  
  300.     if {[catch {::interp eval $slave\
  301.         {source [file join $tcl_library init.tcl]}}\
  302.         msg]} {
  303.         Log $slave "can't source init.tcl ($msg)";
  304.         error "can't source init.tcl into slave $slave ($msg)"
  305.     }
  306.  
  307.     return $slave
  308.     }
  309.  
  310.  
  311.     # Add (only if needed, avoid duplicates) 1 level of
  312.     # sub directories to an existing path list.
  313.     # Also removes non directories from the returned list.
  314.     proc AddSubDirs {pathList} {
  315.     set res {}
  316.     foreach dir $pathList {
  317.         if {[file isdirectory $dir]} {
  318.         # check that we don't have it yet as a children
  319.         # of a previous dir
  320.         if {[lsearch -exact $res $dir]<0} {
  321.             lappend res $dir;
  322.         }
  323.         foreach sub [glob -nocomplain -- [file join $dir *]] {
  324.             if {    ([file isdirectory $sub])
  325.                  && ([lsearch -exact $res $sub]<0) } {
  326.             # new sub dir, add it !
  327.                     lappend res $sub;
  328.                 }
  329.         }
  330.         }
  331.     }
  332.     return $res;
  333.     }
  334.  
  335.     # This procedure deletes a safe slave managed by Safe Tcl and
  336.     # cleans up associated state:
  337.  
  338.     proc ::safe::interpDelete {slave} {
  339.  
  340.         Log $slave "About to delete" NOTICE;
  341.  
  342.     # If the slave has a cleanup hook registered, call it.
  343.     # check the existance because we might be called to delete an interp
  344.     # which has not been registered with us at all
  345.     set hookname [DeleteHookName $slave];
  346.     if {[Exists $hookname]} {
  347.         set hook [Set $hookname];
  348.         if {![::tcl::Lempty $hook]} {
  349.         # remove the hook now, otherwise if the hook
  350.         # calls us somehow, we'll loop
  351.         Unset $hookname;
  352.         if {[catch {eval $hook $slave} err]} {
  353.             Log $slave "Delete hook error ($err)";
  354.         }
  355.         }
  356.     }
  357.  
  358.     # Discard the global array of state associated with the slave, and
  359.     # delete the interpreter.
  360.  
  361.     set statename [InterpStateName $slave];
  362.     if {[Exists $statename]} {
  363.         Unset $statename;
  364.     }
  365.  
  366.     # if we have been called twice, the interp might have been deleted
  367.     # already
  368.     if {[::interp exists $slave]} {
  369.         ::interp delete $slave;
  370.         Log $slave "Deleted" NOTICE;
  371.     }
  372.  
  373.     return
  374.     }
  375.  
  376.     # Set (or get) the loging mecanism 
  377.  
  378. proc ::safe::setLogCmd {args} {
  379.     variable Log;
  380.     if {[llength $args] == 0} {
  381.     return $Log;
  382.     } else {
  383.     if {[llength $args] == 1} {
  384.         set Log [lindex $args 0];
  385.     } else {
  386.         set Log $args
  387.     }
  388.     }
  389. }
  390.  
  391.     # internal variable
  392.     variable Log {}
  393.  
  394.     # ------------------- END OF PUBLIC METHODS ------------
  395.  
  396.  
  397.  
  398.     #
  399.     # sets the slave auto_path to the master recorded value.
  400.     # also sets tcl_library to the first token of the virtual path.
  401.     #
  402.     proc SyncAccessPath {slave} {
  403.     set slave_auto_path [Set [VirtualPathListName $slave]];
  404.     ::interp eval $slave [list set auto_path $slave_auto_path];
  405.     Log $slave \
  406.         "auto_path in $slave has been set to $slave_auto_path"\
  407.         NOTICE;
  408.     ::interp eval $slave [list set tcl_library [lindex $slave_auto_path 0]];
  409.     }
  410.  
  411.     # base name for storing all the slave states
  412.     # the array variable name for slave foo is thus "Sfoo"
  413.     # and for sub slave {foo bar} "Sfoo bar" (spaces are handled
  414.     # ok everywhere (or should))
  415.     # We add the S prefix to avoid that a slave interp called Log
  416.     # would smash our Log variable.
  417.     proc InterpStateName {slave} {
  418.     return "S$slave";
  419.     }
  420.  
  421.     # returns the virtual token for directory number N
  422.     # if the slave argument is given, 
  423.     # it will return the corresponding master global variable name
  424.     proc PathToken {n {slave ""}} {
  425.     if {[string compare "" $slave]} {
  426.         return "[InterpStateName $slave](access_path,$n)";
  427.     } else {
  428.         # We need to have a ":" in the token string so
  429.         # [file join] on the mac won't turn it into a relative
  430.         # path.
  431.         return "p(:$n:)";
  432.     }
  433.     }
  434.     # returns the variable name of the complete path list
  435.     proc PathListName {slave} {
  436.     return "[InterpStateName $slave](access_path)";
  437.     }
  438.     # returns the variable name of the complete path list
  439.     proc VirtualPathListName {slave} {
  440.     return "[InterpStateName $slave](access_path_slave)";
  441.     }
  442.     # returns the variable name of the number of items
  443.     proc PathNumberName {slave} {
  444.     return "[InterpStateName $slave](access_path,n)";
  445.     }
  446.     # returns the staticsok flag var name
  447.     proc StaticsOkName {slave} {
  448.     return "[InterpStateName $slave](staticsok)";
  449.     }
  450.     # returns the nestedok flag var name
  451.     proc NestedOkName {slave} {
  452.     return "[InterpStateName $slave](nestedok)";
  453.     }
  454.     # Run some code at the namespace toplevel
  455.     proc Toplevel {args} {
  456.     namespace eval [namespace current] $args;
  457.     }
  458.     # set/get values
  459.     proc Set {args} {
  460.     eval Toplevel set $args;
  461.     }
  462.     # lappend on toplevel vars
  463.     proc Lappend {args} {
  464.     eval Toplevel lappend $args;
  465.     }
  466.     # unset a var/token (currently just an global level eval)
  467.     proc Unset {args} {
  468.     eval Toplevel unset $args;
  469.     }
  470.     # test existance 
  471.     proc Exists {varname} {
  472.     Toplevel info exists $varname;
  473.     }
  474.     # short cut for access path getting
  475.     proc GetAccessPath {slave} {
  476.     Set [PathListName $slave]
  477.     }
  478.     # short cut for statics ok flag getting
  479.     proc StaticsOk {slave} {
  480.     Set [StaticsOkName $slave]
  481.     }
  482.     # short cut for getting the multiples interps sub loading ok flag
  483.     proc NestedOk {slave} {
  484.     Set [NestedOkName $slave]
  485.     }
  486.     # interp deletion storing hook name
  487.     proc DeleteHookName {slave} {
  488.     return [InterpStateName $slave](cleanupHook)
  489.     }
  490.  
  491.     #
  492.     # translate virtual path into real path
  493.     #
  494.     proc TranslatePath {slave path} {
  495.     # somehow strip the namespaces 'functionality' out (the danger
  496.     # is that we would strip valid macintosh "../" queries... :
  497.     if {[regexp {(::)|(\.\.)} $path]} {
  498.         error "invalid characters in path $path";
  499.     }
  500.     set n [expr [Set [PathNumberName $slave]]-1];
  501.     for {} {$n>=0} {incr n -1} {
  502.         # fill the token virtual names with their real value
  503.         set [PathToken $n] [Set [PathToken $n $slave]];
  504.     }
  505.     # replaces the token by their value
  506.     subst -nobackslashes -nocommands $path;
  507.     }
  508.  
  509.  
  510.     # Log eventually log an error
  511.     # to enable error logging, set Log to {puts stderr} for instance
  512.     proc Log {slave msg {type ERROR}} {
  513.     variable Log;
  514.     if {[info exists Log] && [llength $Log]} {
  515.         eval $Log [list "$type for slave $slave : $msg"];
  516.     }
  517.     }
  518.  
  519.     
  520.     # file name control (limit access to files/ressources that should be
  521.     # a valid tcl source file)
  522.     proc CheckFileName {slave file} {
  523.     # limit what can be sourced to .tcl
  524.     # and forbid files with more than 1 dot and
  525.     # longer than 14 chars
  526.     set ftail [file tail $file];
  527.     if {[string length $ftail]>14} {
  528.         error "$ftail: filename too long";
  529.     }
  530.     if {[regexp {\..*\.} $ftail]} {
  531.         error "$ftail: more than one dot is forbidden";
  532.     }
  533.     if {[string compare $ftail "tclIndex"] && \
  534.         [string compare [string tolower [file extension $ftail]]\
  535.         ".tcl"]} {
  536.         error "$ftail: must be a *.tcl or tclIndex";
  537.     }
  538.  
  539.     if {![file exists $file]} {
  540.         # don't tell the file path
  541.         error "no such file or directory";
  542.     }
  543.  
  544.     if {![file readable $file]} {
  545.         # don't tell the file path
  546.         error "not readable";
  547.     }
  548.  
  549.     }
  550.  
  551.  
  552.     # AliasSource is the target of the "source" alias in safe interpreters.
  553.  
  554.     proc AliasSource {slave args} {
  555.  
  556.     set argc [llength $args];
  557.     # Allow only "source filename"
  558.     # (and not mac specific -rsrc for instance - see comment in ::init
  559.     # for current rationale)
  560.     if {$argc != 1} {
  561.         set msg "wrong # args: should be \"source fileName\""
  562.         Log $slave "$msg ($args)";
  563.         return -code error $msg;
  564.     }
  565.     set file [lindex $args 0]
  566.     
  567.     # get the real path from the virtual one.
  568.     if {[catch {set file [TranslatePath $slave $file]} msg]} {
  569.         Log $slave $msg;
  570.         return -code error "permission denied"
  571.     }
  572.     
  573.     # check that the path is in the access path of that slave
  574.     if {[catch {FileInAccessPath $slave $file} msg]} {
  575.         Log $slave $msg;
  576.         return -code error "permission denied"
  577.     }
  578.  
  579.     # do the checks on the filename :
  580.     if {[catch {CheckFileName $slave $file} msg]} {
  581.         Log $slave "$file:$msg";
  582.         return -code error $msg;
  583.     }
  584.  
  585.     # passed all the tests , lets source it:
  586.     if {[catch {::interp invokehidden $slave source $file} msg]} {
  587.         Log $slave $msg;
  588.         return -code error "script error";
  589.     }
  590.     return $msg
  591.     }
  592.  
  593.     # AliasLoad is the target of the "load" alias in safe interpreters.
  594.  
  595.     proc AliasLoad {slave file args} {
  596.  
  597.     set argc [llength $args];
  598.     if {$argc > 2} {
  599.         set msg "load error: too many arguments";
  600.         Log $slave "$msg ($argc) {$file $args}";
  601.         return -code error $msg;
  602.     }
  603.  
  604.     # package name (can be empty if file is not).
  605.     set package [lindex $args 0];
  606.  
  607.     # Determine where to load. load use a relative interp path
  608.     # and {} means self, so we can directly and safely use passed arg.
  609.     set target [lindex $args 1];
  610.     if {[string length $target]} {
  611.         # we will try to load into a sub sub interp
  612.         # check that we want to authorize that.
  613.         if {![NestedOk $slave]} {
  614.         Log $slave "loading to a sub interp (nestedok)\
  615.             disabled (trying to load $package to $target)";
  616.         return -code error "permission denied (nested load)";
  617.         }
  618.         
  619.     }
  620.  
  621.     # Determine what kind of load is requested
  622.     if {[string length $file] == 0} {
  623.         # static package loading
  624.         if {[string length $package] == 0} {
  625.         set msg "load error: empty filename and no package name";
  626.         Log $slave $msg;
  627.         return -code error $msg;
  628.         }
  629.         if {![StaticsOk $slave]} {
  630.         Log $slave "static packages loading disabled\
  631.             (trying to load $package to $target)";
  632.         return -code error "permission denied (static package)";
  633.         }
  634.     } else {
  635.         # file loading
  636.  
  637.         # get the real path from the virtual one.
  638.         if {[catch {set file [TranslatePath $slave $file]} msg]} {
  639.         Log $slave $msg;
  640.         return -code error "permission denied"
  641.         }
  642.  
  643.         # check the translated path
  644.         if {[catch {FileInAccessPath $slave $file} msg]} {
  645.         Log $slave $msg;
  646.         return -code error "permission denied (path)"
  647.         }
  648.     }
  649.  
  650.     if {[catch {::interp invokehidden\
  651.         $slave load $file $package $target} msg]} {
  652.         Log $slave $msg;
  653.         return -code error $msg
  654.     }
  655.  
  656.     return $msg
  657.     }
  658.  
  659.     # FileInAccessPath raises an error if the file is not found in
  660.     # the list of directories contained in the (master side recorded) slave's
  661.     # access path.
  662.  
  663.     # the security here relies on "file dirname" answering the proper
  664.     # result.... needs checking ?
  665.     proc FileInAccessPath {slave file} {
  666.  
  667.     set access_path [GetAccessPath $slave];
  668.  
  669.     if {[file isdirectory $file]} {
  670.         error "\"$file\": is a directory"
  671.     }
  672.     set parent [file dirname $file]
  673.     if {[lsearch -exact $access_path $parent] == -1} {
  674.         error "\"$file\": not in access_path";
  675.     }
  676.     }
  677.  
  678.     # This procedure enables access from a safe interpreter to only a subset of
  679.     # the subcommands of a command:
  680.  
  681.     proc Subset {slave command okpat args} {
  682.     set subcommand [lindex $args 0]
  683.     if {[regexp $okpat $subcommand]} {
  684.         return [eval {$command $subcommand} [lrange $args 1 end]]
  685.     }
  686.     set msg "not allowed to invoke subcommand $subcommand of $command";
  687.     Log $slave $msg;
  688.     error $msg;
  689.     }
  690.  
  691.     # This procedure installs an alias in a slave that invokes "safesubset"
  692.     # in the master to execute allowed subcommands. It precomputes the pattern
  693.     # of allowed subcommands; you can use wildcards in the pattern if you wish
  694.     # to allow subcommand abbreviation.
  695.     #
  696.     # Syntax is: AliasSubset slave alias target subcommand1 subcommand2...
  697.  
  698.     proc AliasSubset {slave alias target args} {
  699.     set pat ^(; set sep ""
  700.     foreach sub $args {
  701.         append pat $sep$sub
  702.         set sep |
  703.     }
  704.     append pat )\$
  705.     ::interp alias $slave $alias {}\
  706.         [namespace current]::Subset $slave $target $pat
  707.     }
  708.  
  709. }
  710.